<template>
  <view
    v-if="isShow"
    ref="ani"
    class="uni-transition"
    :class="[ani.in]"
    :style="'transform:' + transform + ';' + stylesObject"
    @click="change"
  >
    <slot></slot>
  </view>
</template>

<script>
/* eslint-disable */
// #ifdef APP-NVUE
const animation = uni.requireNativePlugin('animation');
// #endif
/**
 * Transition 过渡动画
 * @description 简单过渡动画组件
 * @tutorial https://ext.dcloud.net.cn/plugin?id=985
 * @property {Boolean} show = [false|true] 控制组件显示或隐藏
 * @property {Array} modeClass = [fade|slide-top|slide-right|slide-bottom|slide-left|zoom-in|zoom-out] 过渡动画类型
 *  @value fade 渐隐渐出过渡
 *  @value slide-top 由上至下过渡
 *  @value slide-right 由右至左过渡
 *  @value slide-bottom 由下至上过渡
 *  @value slide-left 由左至右过渡
 *  @value zoom-in 由小到大过渡
 *  @value zoom-out 由大到小过渡
 * @property {Number} duration 过渡动画持续时间
 * @property {Object} styles 组件样式，同 css 样式，注意带’-‘连接符的属性需要使用小驼峰写法如：`backgroundColor:red`
 */
export default {
  name: 'uniTransition',
  props: {
    show: {
      type: Boolean,
      default: false,
    },
    modeClass: {
      type: Array,
      default() {
        return [];
      },
    },
    duration: {
      type: Number,
      default: 300,
    },
    styles: {
      type: Object,
      default() {
        return {};
      },
    },
  },
  data() {
    return {
      isShow: false,
      transform: '',
      ani: { in: '', active: '' },
    };
  },
  watch: {
    show: {
      handler(newVal) {
        if (newVal) {
          this.open();
        } else {
          this.close();
        }
      },
      immediate: true,
    },
  },
  computed: {
    stylesObject() {
      let styles = {
        ...this.styles,
        'transition-duration': this.duration / 1000 + 's',
      };
      let transfrom = '';
      for (let i in styles) {
        let line = this.toLine(i);
        transfrom += line + ':' + styles[i] + ';';
      }
      return transfrom;
    },
  },
  created() {
    // this.timer = null
    // this.nextTick = (time = 50) => new Promise(resolve => {
    // 	clearTimeout(this.timer)
    // 	this.timer = setTimeout(resolve, time)
    // 	return this.timer
    // });
  },
  methods: {
    change() {
      this.$emit('click', {
        detail: this.isShow,
      });
    },
    open() {
      clearTimeout(this.timer);
      this.isShow = true;
      this.transform = '';
      this.ani.in = '';
      for (let i in this.getTranfrom(false)) {
        if (i === 'opacity') {
          this.ani.in = 'fade-in';
        } else {
          this.transform += `${this.getTranfrom(false)[i]} `;
        }
      }
      this.$nextTick(() => {
        setTimeout(() => {
          this._animation(true);
        }, 50);
      });
    },
    close(type) {
      clearTimeout(this.timer);
      this._animation(false);
    },
    _animation(type) {
      let styles = this.getTranfrom(type);
      // #ifdef APP-NVUE
      if (!this.$refs['ani']) return;
      animation.transition(
        this.$refs['ani'].ref,
        {
          styles,
          duration: this.duration, //ms
          timingFunction: 'ease',
          needLayout: false,
          delay: 0, //ms
        },
        () => {
          if (!type) {
            this.isShow = false;
          }
          this.$emit('change', {
            detail: this.isShow,
          });
        },
      );
      // #endif
      // #ifndef APP-NVUE
      this.transform = '';
      for (let i in styles) {
        if (i === 'opacity') {
          this.ani.in = `fade-${type ? 'out' : 'in'}`;
        } else {
          this.transform += `${styles[i]} `;
        }
      }
      this.timer = setTimeout(() => {
        if (!type) {
          this.isShow = false;
        }
        this.$emit('change', {
          detail: this.isShow,
        });
      }, this.duration);
      // #endif
    },
    getTranfrom(type) {
      let styles = {
        transform: '',
      };
      this.modeClass.forEach(mode => {
        switch (mode) {
          case 'fade':
            styles.opacity = type ? 1 : 0;
            break;
          case 'slide-top':
            styles.transform += `translateY(${type ? '0' : '-100%'}) `;
            break;
          case 'slide-right':
            styles.transform += `translateX(${type ? '0' : '100%'}) `;
            break;
          case 'slide-bottom':
            styles.transform += `translateY(${type ? '0' : '100%'}) `;
            break;
          case 'slide-left':
            styles.transform += `translateX(${type ? '0' : '-100%'}) `;
            break;
          case 'zoom-in':
            styles.transform += `scale(${type ? 1 : 0.8}) `;
            break;
          case 'zoom-out':
            styles.transform += `scale(${type ? 1 : 1.2}) `;
            break;
        }
      });
      return styles;
    },
    _modeClassArr(type) {
      let mode = this.modeClass;
      if (typeof mode !== 'string') {
        let modestr = '';
        mode.forEach(item => {
          modestr += item + '-' + type + ',';
        });
        return modestr.substr(0, modestr.length - 1);
      } else {
        return mode + '-' + type;
      }
    },
    // getEl(el) {
    // 	console.log(el || el.ref || null);
    // 	return el || el.ref || null
    // },
    toLine(name) {
      return name.replace(/([A-Z])/g, '-$1').toLowerCase();
    },
  },
};
</script>

<style>
.uni-transition {
  transition-duration: 0.3s;
  transition-property: transform, opacity;
  transition-timing-function: ease;
}
.fade-in {
  opacity: 0;
}
.fade-active {
  opacity: 1;
}
.slide-top-in {
  /* transition-property: transform, opacity; */
  transform: translateY(-100%);
}
.slide-top-active {
  transform: translateY(0);
  /* opacity: 1; */
}
.slide-right-in {
  transform: translateX(100%);
}
.slide-right-active {
  transform: translateX(0);
}
.slide-bottom-in {
  transform: translateY(100%);
}
.slide-bottom-active {
  transform: translateY(0);
}
.slide-left-in {
  transform: translateX(-100%);
}
.slide-left-active {
  opacity: 1;
  transform: translateX(0);
}
.zoom-in-in {
  transform: scale(0.8);
}
.zoom-out-active {
  transform: scale(1);
}
.zoom-out-in {
  transform: scale(1.2);
}
</style>
